/******************************************************************************* * Copyright (c) 2000, 2010 IBM Corporation and others. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * IBM Corporation - initial API and implementation * Nikolay Botev - bug 240651 *******************************************************************************/ package org.eclipse.ui.part; import org.eclipse.core.runtime.IProgressMonitor; import org.eclipse.swt.widgets.Composite; import org.eclipse.ui.IEditorInput; import org.eclipse.ui.IEditorPart; import org.eclipse.ui.IEditorReference; import org.eclipse.ui.IEditorSite; import org.eclipse.ui.IPartListener2; import org.eclipse.ui.IPropertyListener; import org.eclipse.ui.IWorkbenchPart; import org.eclipse.ui.IWorkbenchPartReference; import org.eclipse.ui.PartInitException; import org.eclipse.ui.internal.EditorSite; import org.eclipse.ui.internal.PartService; import org.eclipse.ui.internal.PartSite; import org.eclipse.ui.internal.WorkbenchPage; /** * A AbstractMultiEditor is a composite of editors. * * This class is intended to be subclassed. * * @since 1.4 */ public abstract class AbstractMultiEditor extends EditorPart { private int activeEditorIndex; private IEditorPart innerEditors[]; private IPartListener2 propagationListener; /** * Constructs an editor to contain other editors. */ public AbstractMultiEditor() { super(); } /** * Handles a property change notification from a nested editor. The default * implementation simply forwards the change to listeners on this multi * editor by calling <code>firePropertyChange</code> with the same property * id. For example, if the dirty state of a nested editor changes (property * id <code>ISaveablePart.PROP_DIRTY</code>), this method handles it by * firing a property change event for <code>ISaveablePart.PROP_DIRTY</code> * to property listeners on this multi editor. * <p> * Subclasses may extend or reimplement this method. * </p> * * @param propId * the id of the property that changed * @since 1.4 */ protected void handlePropertyChange(int propId) { firePropertyChange(propId); } /* * @see IEditorPart#doSave(IProgressMonitor) */ public void doSave(IProgressMonitor monitor) { for (int i = 0; i < innerEditors.length; i++) { IEditorPart e = innerEditors[i]; e.doSave(monitor); } } /* * @see IEditorPart#doSaveAs() */ public void doSaveAs() { //no-op } /* * @see IEditorPart#init(IEditorSite, IEditorInput) */ public void init(IEditorSite site, IEditorInput input) throws PartInitException { init(site, (MultiEditorInput) input); } /** * @param site * @param input * @throws PartInitException * * @see IEditorPart#init(IEditorSite, IEditorInput) */ public void init(IEditorSite site, MultiEditorInput input) throws PartInitException { setInput(input); setSite(site); setPartName(input.getName()); setTitleToolTip(input.getToolTipText()); setupEvents(); } /* * @see IEditorPart#isDirty() */ public boolean isDirty() { for (int i = 0; i < innerEditors.length; i++) { IEditorPart e = innerEditors[i]; if (e.isDirty()) { return true; } } return false; } /* * @see IEditorPart#isSaveAsAllowed() */ public boolean isSaveAsAllowed() { return false; } /* * @see IWorkbenchPart#setFocus() */ public void setFocus() { innerEditors[activeEditorIndex].setFocus(); } /** * Returns the active inner editor. * @return the active editor */ public final IEditorPart getActiveEditor() { return innerEditors[activeEditorIndex]; } /** * Returns an array with all inner editors. * @return the inner editors */ public final IEditorPart[] getInnerEditors() { return innerEditors; } /** * Set the inner editors. * * Should not be called by clients. * * @param children * the inner editors of this multi editor * @noreference This method is not intended to be referenced by clients. */ public final void setChildren(IEditorPart[] children) { innerEditors = children; activeEditorIndex = 0; for (int i = 0; i < children.length; i++) { children[i].addPropertyListener( new IPropertyListener() { public void propertyChanged(Object source, int propId) { handlePropertyChange(propId); } }); } innerEditorsCreated(); } /** * Called as soon as the inner editors have been created and are available. */ protected abstract void innerEditorsCreated(); /** * Activates the given nested editor. * * @param part the nested editor * @since 1.4 */ public void activateEditor(IEditorPart part) { activeEditorIndex = getIndex(part); IEditorPart e = getActiveEditor(); EditorSite innerSite = (EditorSite) e.getEditorSite(); ((WorkbenchPage) innerSite.getPage()).requestActivation(e); } /** * Returns the index of the given nested editor. * * @return the index of the nested editor * @since 1.4 */ protected int getIndex(IEditorPart editor) { for (int i = 0; i < innerEditors.length; i++) { if (innerEditors[i] == editor) { return i; } } return -1; } /** * Set up the AbstractMultiEditor to propagate events like partClosed(). * * @since 1.4 */ private void setupEvents() { propagationListener = new IPartListener2() { public void partActivated(IWorkbenchPartReference partRef) { } public void partBroughtToTop(IWorkbenchPartReference partRef) { } public void partClosed(IWorkbenchPartReference partRef) { IWorkbenchPart part = partRef.getPart(false); if (part == AbstractMultiEditor.this && innerEditors != null) { PartService partService = ((WorkbenchPage) getSite() .getPage()).getPartService(); for (int i = 0; i < innerEditors.length; i++) { IEditorPart editor = innerEditors[i]; IWorkbenchPartReference innerRef = ((PartSite) editor .getSite()).getPartReference(); partService.firePartClosed(innerRef); } } } public void partDeactivated(IWorkbenchPartReference partRef) { } public void partOpened(IWorkbenchPartReference partRef) { IWorkbenchPart part = partRef.getPart(false); if (part == AbstractMultiEditor.this && innerEditors != null) { PartService partService = ((WorkbenchPage) getSite() .getPage()).getPartService(); for (int i = 0; i < innerEditors.length; i++) { IEditorPart editor = innerEditors[i]; IWorkbenchPartReference innerRef = ((PartSite) editor .getSite()).getPartReference(); partService.firePartOpened(innerRef); } } } public void partHidden(IWorkbenchPartReference partRef) { } public void partVisible(IWorkbenchPartReference partRef) { } public void partInputChanged(IWorkbenchPartReference partRef) { } }; getSite().getPage().addPartListener(propagationListener); } /** * Release the added listener. * * @since 1.4 */ public void dispose() { getSite().getPage().removePartListener(propagationListener); super.dispose(); } /** * This method is called after createPartControl has been executed and * should return the container for the given inner editor. * * @param innerEditorReference * a reference to the inner editor that is being created. * @return the container in which the inner editor's pane and part controls * are to be created. */ public abstract Composite getInnerEditorContainer(IEditorReference innerEditorReference); }